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Using a TMS320C30 Serial Port as an 


Asynchronous RS-232 Port 


Abstract 


Although the TMS320C30 serial ports were designed to be used as 
synchronous ports, they can be used as asynchronous ports under 
software control. This application note describes the hardware and 
software to use a TMS320C30 serial port as an asynchronous port. 
A schematic diagram and a lengthy code listing are provided to 


illustrate the solution. 
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Design Problem 


Solution 


SPRA240 


Although the TMS320C30 serial ports were designed to be used as 
synchronous ports, they can be used as asynchronous ports with a 
little creative software. This application note describes the hardware 
and software to use a TMS320C30 serial port as an asynchronous 
port. 


How it works 


This design relies on the fact that received RS-232 signals always 
start with a “start bit” that is not part of the data and end with one or 
more “stop bits” that are also not part of the data. This design keeps 
the receiver turned off and an interrupt (also tied to the receive line) 
turned on when not receiving a character. When the interrupt goes 
off, this signals a start bit on the line. The code then turns the 
interrupt off and the receiver on; the data comes in as a normal 8-bit 
character. The stop bits assure the TMS320C30 has time to handle 
the data before the next character. 


The transmitter basically frames the data into 16-bit words, adding a 
start bit, the character to send, and the stop bits. This will result in 
up to 6 clock cycles (RS-232 clock rate) where bandwidth on the 
channel is “wasted.” (Think of it as having 7 stop bits. That’s kind of 
how it works.) A more efficient (but more complicated) design could 
be done, but was not necessary for my project. Characters go in and 
out the serial port “backwards” from the RS-232 method; they must 
be bit-swapped to be correct. 


The serial port is set up as a Continuous transmit normal port. Frame 
syncs are not used and internal clocks are used for the serial port 
timing; these are timed of the TMS320C30 clock and need to be 
adjusted if clock rates change. 


Hardware 


Little hardware design needs to be done to handle this, basically just 
wire the serial port to the TMS320C30 properly. 


Figure 1. Schematic Diagram 


75C189A (or equiv.) 


> 


75C188 (or equiv.) 
TXD 


Notice that the received signal is tied to DRx (Data Receive) and 
also INTx (any of the TMS320C30 interrupts). DXx (Data 
TRANSmit) must be pulled high to avoid having to have the 
TMS320C30 constantly supply “1” on the line when it has nothing to 
transmit. The clocks and frame syncs are not used. 


Software 


The real meat of this design lies in the software. It must handle the 
interrupts and port setups and the queuing of the data. Actual code 
and descriptions follow in this article. The code ran under an 
operating system written by me, but the operation of the OS routines 
should be obvious. 


Transmitter 


The transmitter does not do very much; just frames the data 
properly, waits for the transmitter to be free, and sends the data. 
The transmitter interrupt also drove the OS timer tick; therefore the 
transmitter was constantly driven with data even when idle. 


Receiver 
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The receiver does a lot more than the transmitter; some interrupt 
tricks supply the necessary “sync to async” conversion. Normally the 
serial port receiver is turned off. An interrupt comes in (the 
rec_coming interrupt) when a start bit comes in the receiver. This 
will turn off the rec_coming interrupt and start the receiver. The next 
8 bits coming in the serial port should be the character desired. After 
these 8 bits are received the recO interrupt goes off. This will handle 
the received character; turn off the receiver, and turn back on the 
rec_coming interrupt to wait for the next start bit. 


Example 1. Code Listing 


[KKK RK KK KK KK A IA A A A I I / 


/* 


* io.c - The I/O routines and tasks to handle I/O to the C30 serial port. 


a) 


#include “monitor.h” 
#include “debug.h” 


#include “io.h” 


Queue_Id 
Queue_Id 
Queue_Id 
int 


static Queue_Id 
static Queue_Id 


/* 


gets queue; 
rec_int_queue; 
io_state; 
rec_ready; 


wait_rec_int[2]; 
wait_rec_cmd[2]; 


* invert_8 - swaps the bits in the 8 bit character supplied. 


ue 


char invert_8 (inchar) 


char inchar; 


{ 


char outchar; 


outchar = 0; 


if (inchar & 0x01) 


{ 


outchar |= 0x80; 


if (inchar & 0x02) 


outchar |= 0x40; 


if (inchar & 0x04) 


outchar |= 0x20; 


if (inchar & 0x08) 


outchar |= 0x10; 


if (inchar & 0x10) 


outchar |= 0x08; 


if (inchar & 0x20) 
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outchar |= 0x04; 
} 
if (inchar & 0x40) 
{ 
outchar |= 0x02; 
} 
if (inchar & 0x80) 
{ 
outchar |= 0x01; 
} 
return (outchar) ; 
} 
/* 
* The get string task. This receives request to receive strings then 
* receives them and sends the result back to the requesting task. 
xf 
void 
gets_task() 
{ 
unsigned int my_tid; 
unsigned int msg; 
unsigned int tid; 
unsigned int qid; 
unsigned int dummyl; 
unsigned int dummy2; 
Buffer_Id buf; 
void *bufptr; 


char outbuf[3]; 


char *out_loc; 
unsigned int count; 
unsigned int max_size; 
int finished; 
unsigned int c; 


io_state = NODEBUG_STATE 


~ 


my_tid = 0; 


os_task_inquiry(&my_tid, NULL); /* Get my task id (and therefore my) */ 
/* main queue id. */ 


os_create_queue (&égets_queue) ; /* Create another queue for requests */ 
/* to get data. */ 
rec_int_queue = my_tid; /* My main queue same as tid */ 
wait_rec_int[0] = rec_int_queue; /* Set up queue lists for wait queues * / 
wait_rec_int [1] = END_QUEUE; 
wait_rec_cmd[0] = gets_queue; 
wait_rec_cmd[1] = END_QUEUE; 
while (TRUE) 
{ 
rec_ready = FALSE; /* Not receiving any data here */ 


/* Wait for someone to request a string */ 
os_wait_fetch(wait_rec_cmd, é&msg, &buf, &bufptr, &tid, &qid); 
if (buf != NO_BUFFER) 


os_free_buffer (buf); 
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rec_ready = TRUE; /* Now we are receiving data */ 
/* 
* The following is not 32-bit clean, but it doesn’t matter for 
# C308 
*/ 
max_size = (msg > 24) & Oxff; /* Get the num bytes to receive */ 


out_loc = ((char *) (msg & Oxffffff)); /* Get the address to put */ 


/* the string in. */ 
count = 0; 
finished = FALSE; 
while (!finished) 
{ 


if (count = max_size) /* If all the data is in, send a msg */ 


{ /* back to the requestor */ 
*out_loc = ’\0'; 
finished = TRUE; 
os_put_queue (REC_FINISHED, NO_BUFFER, tid); 


} 


else 


{ 


/* Wait for the receiver to send me some data */ 


os_wait_fetch(wait_rec_int, é&c, &buf, &bufptr, &dummyl, 


if (buf != NO_BUFFER) 
{ 


os_free_buffer (buf); 
} 


outbuf[0] = c; /* Put the received data into a buf */ 
outbuf[1] = ’\0’; /* so it can be echoed. */ 
puts (outbuf); /* Echo the data */ 
if (c == ’\n’) /* If a newline is received, finish */ 
{ /* the receive. */ 

*out_loc = ’\0’'; 


finished = TRUE; 
os_put_queue (REC_FINISHED, NO_BUFFER, tid); 


} 
else /* else put the character into the */ 
{ /* buffer. */ 

*out_loc = c; 

out_loctt+; 

count++; 


} 
/* 


édummy 2) ; 


* Receive handler. This routine is called by the interrupt handler that 


* is called when a byte is received from the com port. 
“ar 

void 

rec_hndl () 

{ 


int rec_char; 
regioncount = 1; 


/* Data from RS-232 is backwards, flip it around */ 
rec_char = invert_8((*RECLOC) & Oxff); 
if (rec_char == 0x0d) /* Map ctrl-m to newline (No raw mode!) 


{ 


rec_char = '\n’; 
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} 
if (io0_state == DEBUG_STATE) /* If the debugger is on, send all */ 
{ /* data to it. */ 

os_put_queue (rec_char, NO_BUFFER, debug_q) ; 


} 
else if (rec_char == 0x03) /* A ctrl-c activates the debugger. */ 
{ 


io_state = DEBUG_STATE; 
os_start_task (debug_tid) ; 


lse if (rec_ready) /* Send data to the gets task if it wants it. */ 


os_put_queue (rec_char, NO_BUFFER, rec_int_queue) ; 


} 


regioncount = 0; 
} 
#define XMTLOC ((int *) 0x808048) 
#define RECLOC ((int *) Ox80804c) 
#define XMT_PRT_CTL ((int *) 0x808042) 


Queue_Id puts_queue; 
Queue_Id xmt_int_queue; 
static Queue_Id wait_xmt_int [2]; 
static Queue_Id wait_xmt_cmd[2]; 
int xmt_data; 

/* 


* The put string routine. This task will put strings out to the serial port. 
ny 
void 
puts (string) 
char *string; 


{ 


Task_Id my_tid; 

Queue_Id wait_fini[2]; 

unsigned int msg; 

Task_Id tid; 

Queue_Id gid; 

Buffer_Id buf; 

void *bufptr; 

my_tid = 0; 

os_task_inquiry(&my_tid, NULL); /* Get my task id. */ 

wait_fini[0] = my_tid; /* Use my task id as the queue to */ 


wait_fini[1] 


END_QUEUE; /* receive xmit ready messages. */ 


/* Send a pointer to the string to the transmit task. */ 
os_put_queue((unsigned int) string, NO_BUFFER, puts_queue) ; 


/* Wait for it to respond. */ 

os_wait_fetch(wait_fini, &msg, &buf, &bufptr, &tid, &qid); 
if (buf != NO_BUFFER) 

{ 


os_free_buffer (buf); 
} 
/* Ignore all messages that are not a send finished from the xmit task */ 
while (msg != SEND_FINISHED) 
{ 
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} 
/* 


os_wait_fetch(wait_fini, &msg, &buf, &bufptr, &tid, &qid); 
if (buf != NO_BUFFER) 
{ 


os_free_buffer (buf); 
} 


* The put string task. This task will wait for strings on its input queue 
* and transmit them to the serial port. 


*f 
puts_task () 
{ 
Task_Id my_tid; 
char *msg; 
Task_Id tid; 
Queue_Id gid; 
unsigned int dummyl, dummy2, dummy3; 
int newline_flag; 
Buffer_Id buf; 
void AHuLpEL; 
my_tid = 0; 
os_task_inquiry(&my_tid, NULL); /* Get my task id. */ 
os_create_queue (&puts_queue) ; /* Create a queue to get send requests*/ 
xmt_int_queue = my_tid; /* My queue to get transmitter */ 
/* interrupt messages. */ 
wait_xmt_int[0] = xmt_int_queue; /* Set up receive queues. */ 
wait_xmt_int [1] = END_QUEUE; 
wait_xmt_cmd[0] = puts_queue; 
wait_xmt_cmd[1] = END_QUEUE; 
newline_flag = FALSE; 


xmt_data = FALSE; 


while (TRUE) 


{ 


&qid); 


/* Wait for a string to transmit. */ 
os_wait_fetch(wait_xmt_cmd, (unsigned int *) &msg, &buf, &bufptr, &tid, 


if (buf != NO_BUFFER) 
{ 


os_free_buffer (buf); 
} 


/* 
* Ok, now I am transmitting. Wait for the transmitter to tell me 
* that I can send some data. 
*/ 
xmt_data = TRUE; 
os_wait_fetch(wait_xmt_int, édummyl, &buf, &bufptr, &dummy2, &dummy3) ; 
if (buf != NO_BUFFER) 
{ 


os_free_buffer (buf); 
: 
/* 


* Send the whole message. Make sure to send the last new line 
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* even if currently pointing to the EOS character. 


x]: 
while ((*msg != ’\0’) || (newline_flag) ) 
{ 
if (newline_flag) /* Tf transmitting a newline, (ctrl-j), also */ 
{ /* send a cariage return (ctrl-m). */ 
*XMTLOC = (( (int) invert_8((char)0x0d)) & Oxfeff) | Oxfe00; 


newline_flag = FALSE; 
} 


else 


{ 


if (*msg == '\n’) /* If a newline, set up to senda */ 
{ /* ctrl-m next. * 
newline_flag = TRUE; 


} 


/* 
* Put the character into the output buffer. The first 
* 7 bits are transmitted as 1, the next is the start bit, 
* the rest is the character. 
*/ 
*XMTLOC = (((int) invert_8(*msg)) & Oxfeff) | Oxfe00; 
msgt+; 
} 
/*Wait for the transmitter to tell me I can send the next char*/ 
os_wait_fetch (wait_xmt_int, édummyl1, &buf, &bufptr, &dummy2, &édummy3) ; 
if (buf != NO_BUFFER) 
{ 


os_free_buffer (buf); 
} 
} 
xmt_data = FALSE; /* No longer receiveing data. */ 
*XMTLOC = Oxffff; /* Prime the transmitter to send ones. */ 


/* Inform the requestor that the send is finished. */ 
os_put_queue (SEND_FINISHED, NO_BUFFER, tid); 


Nw 


* 
* transmit interrupt handler. This routine is called whenever the transmit 
* interrupt for the serial port goes off. It continuously keeps the 
* transmitter primed because the transmit interrupt is also used as the 
* clock interrupt. 
lear 
void 
xmt_hnd1l () 
{ 
if (xmt_data) /* If the puts task is waiting interrupt info... */ 
{ 
regioncount = 1; /* Interrupts should already be turned off, */ 
/* set the critical region count to */ 
/* veflect that. */ 
pe 
* Send a message to the puts task to tell it to send the next 
* char. If the send fails, go ahead and prime the transmitter. 
ef 
if (os_put_queue (0, NO_BUFFER, xmt_int_queue) != 0) 
{ 
*XMTLOC = Oxffff; 
} 
regioncount = 0; 
} 
else /* If the puts task is not sending, prime the transmitter */ 
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*XMTLOC = Oxffff; 
} 


[KKK KK RK KK RK A A A A IA A A A I / 


;ioasm.asm —- the assembly langage support routines for I/O handling for the 
i; C30 serial port. 


-global xmt0 
-global rec0 


-global rec_coming 

-global _init_io 

-global _rec_hndl 

-global _xmt_hndl 

-global _os_tick, save_task, restore_task 
;* xmtO -— handle an interrupt from the serial port transmitter. This also 
Hees calls the OS tick routine. Note that the save and restore tasks 
aad are called because this can result in a task switch. 

- text 
xmt0 

CALL save_task 

CALL _xmt_hndl 

CALL _os_tick 

CALL restore_task 

RETI 
rec_ser_cnt .word 808040h ; Address of serial port status register 
s_recc_int .word 000000002h ; Mask for the receive interrupt 
c_recc_int .word OffffLffL£dh ; Inverted mask to clear the rec int. 
reset_rec .word Of7f£ffLffFfh ; Mask to write a 0 to the rec reset 


unreset_rec .word 008000000h ; Unreset the receiver. 


7* recO —- Handle an interrupt from the serial port receiver to inform it 
ie of the receipt of a byte on the serial port. This routine will 
ies turn off the receiver and restore the interrupt telling it that 
ne a byte is about to come. 

rec0 


CALL save_task 


LDP @rec_ser_cnt,DP 

LDI @rec_ser_cnt,ARO 

LDI *+ARO (0) ,RO ;Reset the receiver 

AND @reset_rec, RO 

STI RO, *+ARO (0) 

AND @c_recc_int, IF ;clear receive coming interrupt 

OR @s_recc_int, IE ;enable the interrupt for the next byte 

CALL _rec_hndl 

CALL restore_task 

RETI 
;* rec_coming - This interrupt handle is called by INT1. It is tied to the 
ea receive data line, it will be called when the start bit is received 
a for a character of information. It will turn on the serial receiver 
;* (and the serial receiver interrupt) and turn its own interrupt off. 
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rec_coming 


PUSH ST 
PUSH RO 
PUSH DP 
PUSH ARO 
LDP @rec_ser_cnt,DP 
LDI @rec_ser_cnt,ARO 
AND @c_recc_int, IE ;do not allow the receive coming interrupt 
LDI *+ARO (0) ,RO ;Ready the receiver 
OR @unreset_rec, RO 
Sri RO, *+ARO (0) 
POP ARO 
POP DP 
POP RO 
POP OL 
RETI 
gl_prt_cnt .word 0068400c4h ; Initial setup for the serial port 
; status register. This sets the 
; following things: 
- FSX is output. 
4 Fixed data rate signalling 
: Standard frame sync mode 
; Internal xmit clk 
; Internal rec clk 
7 Active high DX and DR 
; XLEN - 16 bits 
: RLEN - 8 bits 
7 Transmitter interrupt enabled 
H Receive interrupt enabled 
° Activate the transmitter 
7 Deactivate the receiver 
x_prt_ent .word 000000111h ; Setup for the transmit port control 
; register. Set all the transmit 
; pins as serial port pins. 
r_prt_ent .word 000000111h ; Setup for the receive port control 
; register. Set all the receive 
; pins as serial port pins. 
tmr_cnt -word 0000003cfh ; Setup for the timer control reg. 
7 Starts the timer 
: Free run the timer (no hold) 
; Clock mode 
; Internal clock source 
tmr_per .word 00434042Ah ; Timer periods. This is 1076 for 
; the receiver, which is a little 
; slow. This makes sure we don’t 
; shift in time before the bits. 
; These also assume a 20.48MHZ clock 
; in the C30; these values will have 
; to be adjusted for different clock 
7 rates. 
enab_int -word 000000032h ; Enable serial xmit, serial recieve, 


; and int 1 for serial port stuff. 
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first_xmt 


_init_io 
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.word 


ooo0d0ffffh 
@rec_ser_cnt,DP 
@rec_ser_cnt,ARO 


@x_prt_cnt,RO 
RO, *+ARO (2) 


@r_prt_cnt,RO 
RO, *+ARO (3) 


@tmr_per, RO 
RO, *+ARO (6) 


@tmr_cnt, RO 
RO, *+ARO (4) 


@gl_prt_cnt, RO 
RO, *+ARO (0) 


@enab_int, IE 


@first_xmt, RO 
RO, *+ARO (8) 


Set up the transmit control port. 


Set up the receive control port. 


Set the timer period register. 


Set the timer control register. 


set 


ct 


he global serial control register. 


Enable interrupts. 


Start the transmitter sending 1’s 


